    <!DOCTYPE html>
<html lang="zh-cn">
	<head>
		<meta charset="utf-8">
		<meta http-equiv="X-UA-Compatible" content="IE=edge,chrome=1">
		<meta name="viewport" content="width=device-width, initial-scale=1">
		
		<meta name="description" content="Nodejh&#39;s Blog">
		<meta name="generator" content="Hugo 0.31.1" />
		<title>在 Oracle 中设置自增列 &middot; node</title>
		<link rel="shortcut icon" href="http://nodejh.com/images/favicon.ico">
		<link rel="stylesheet" href="http://nodejh.com/css/style.css">
		<link rel="stylesheet" href="http://nodejh.com/css/highlight.css">
		

		
		<link rel="stylesheet" href="http://nodejh.com/css/font-awesome.min.css">
		

		
		<link href="http://nodejh.com/index.xml" rel="alternate" type="application/rss+xml" title="node" />
		
	</head>

    <body>
       <nav class="main-nav">
	
	
		<a href='http://nodejh.com/'> <span class="arrow">←</span>Home</a>
	
	<a href='http://nodejh.com/post'>Archive</a>
	<a href='http://nodejh.com/tags'>Tags</a>
	<a href='http://nodejh.com/projects'>Projects</a>
	<a href='http://nodejh.com/about'>About</a>

	

	
	<a class="cta" href="http://nodejh.com/index.xml">Subscribe</a>
	
</nav>


        <section id="wrapper" class="post">
            <article>
                <header>
                    <h1>
                        在 Oracle 中设置自增列
                    </h1>
                    <h2 class="headline">
                    Feb 20, 2017 20:26
                    · 1676 words
                    · 4 minutes read
                      <span class="tags">
                      
                      
                          
                              <a href="http://nodejh.com/tags/oracle">Oracle</a>
                          
                              <a href="http://nodejh.com/tags/%E6%95%B0%E6%8D%AE%E5%BA%93">数据库</a>
                          
                              <a href="http://nodejh.com/tags/database">Database</a>
                          
                      
                      
                      </span>
                    </h2>
                </header>
                
                  
                    <div id="toc">
                      <nav id="TableOfContents">
<ul>
<li>
<ul>
<li><a href="#一-什么是自增列">一、什么是自增列 ？</a></li>
<li><a href="#二-在-oracle-11g-中设置自增字段">二、在  Oracle 11g 中设置自增字段</a>
<ul>
<li><a href="#1-创建表">1. 创建表</a></li>
<li><a href="#2-创建序列">2. 创建序列</a></li>
<li><a href="#3-创建触发器">3. 创建触发器</a></li>
<li><a href="#4-一些值得注意的地方">4. 一些值得注意的地方</a>
<ul>
<li><a href="#4-1-插入指定-id">4.1 插入指定 ID</a></li>
<li><a href="#4-2-字段加引号">4.2 字段加引号</a></li>
<li><a href="#4-3-squence">4.3 SQUENCE</a></li>
<li><a href="#4-4-性能">4.4 性能</a></li>
</ul></li>
</ul></li>
<li><a href="#三-在-oracle-12c-中设置自增字段">三、在 Oracle 12c 中设置自增字段</a></li>
<li><a href="#四-总结">四、总结</a></li>
</ul></li>
</ul>
</nav>
                    </div>
                  
                
                <section id="post-body">
                    <p>如果你经常使用 MySQL，你肯定对 <code>AUTO_INCREMENT</code> 非常熟悉，因为经常要用到它。</p>

<p></p>

<h2 id="一-什么是自增列">一、什么是自增列 ？</h2>

<p>自增列是数据库中值随插入的每个行自动增加的一列。它最常用于主键或 ID 字段，这样每次增加一行时，不用指该字段的值，它就会自动增加，而且是唯一的。</p>

<p>当在 MySQL 中定义列时，我们可以指定一个名为 <code>AUTO_INCREMENT</code> 的参数。然后，每当将新值插入此表中时，放入此列的值比最后一个值加 <code>1</code>。</p>

<p>但很不幸，Oracle 没有 <code>AUTO_INCREMENT</code> 功能。 那要如何在Oracle中做到这一点呢？</p>

<h2 id="二-在-oracle-11g-中设置自增字段">二、在  Oracle 11g 中设置自增字段</h2>

<h3 id="1-创建表">1. 创建表</h3>

<p>首先创建一张用于测试的表：</p>

<pre><code>CREATE TABLE &quot;TEST&quot; (
    ID NUMBER(11) PRIMARY KEY,
    NAME VARCHAR2(50BYTE) NOT NULL
);
</code></pre>

<h3 id="2-创建序列">2. 创建序列</h3>

<p>然后创建一个名为 <code>TEST_ID_SEQ</code> 的序列（序列名称自己随意设定）：</p>

<pre><code>CREATE SEQUENCE TEST_ID_SEQ
INCREMENT BY 1
START WITH 100
MAXVALUE 999999999
NOCYCLE
NOCACHE;
</code></pre>

<p>如果要删除序列，可以使用下面的 SQL 命令：</p>

<pre><code>DROP SEQUENCE TEST_ID_SEQ;
</code></pre>

<p>对 <code>SEQUENCE</code> 的一些说明：</p>

<ul>
<li><code>INCREMENT BY</code> 用于指定序列增量（默认值：1），如果指定的是正整数，则序列号自动递增，如果指定的是负数，则自动递减。</li>
<li><code>START WITH</code> 用于指定序列生成器生成的第一个序列号，当序列号顺序递增时默认值为序列号的最小值，当序列号顺序递减时默认值为序列号的最大值。</li>
<li><code>MAXVALUE</code> 用于指定序列生成器可以生成的组大序列号（必须大于或等于 <code>START WITH</code>，并且必须大于 <code>MINVALUE</code>），默认为 <code>NOMAXVALUE</code>。</li>
<li><code>MINVALUE</code> 用于指定序列生成器可以生成的最小序列号（必须小于或等于 <code>START WITH</code>，并且必须小于 <code>MAXVALUE</code>），默认值为 <code>NOMINVALUE</code>。</li>
<li><code>CYCLE</code> 用于指定在达到序列的最大值或最小值之后是否继续生成序列号，默认为 <code>NOCYCLE</code>。</li>
<li><code>CACHE</code> 用于指定在内存中可以预分配的序列号个数（默认值：20）。</li>
</ul>

<p>到这一步其实就已经可以实现字段自增，只要插入的时候，将 ID 的值设置为序列的下一个值 <code>TEST_ID_SEQ.NEXTVAL</code> 就可以了：</p>

<pre><code>SQL&gt; INSERT INTO &quot;TEST&quot; (&quot;ID&quot;, &quot;NAME&quot;) VALUES (TEST_ID_SEQ.NEXTVAL, 'name1');
SQL&gt; INSERT INTO &quot;TEST&quot; (&quot;ID&quot;, &quot;NAME&quot;) VALUES (TEST_ID_SEQ.NEXTVAL, 'name2');
SQL&gt; INSERT INTO &quot;TEST&quot; (&quot;ID&quot;, &quot;NAME&quot;) VALUES (TEST_ID_SEQ.NEXTVAL, 'name3');
SQL&gt; SELECT * FROM &quot;TEST&quot;;

ID   NAME
---  ------
100	name1
101	name2
102	name3

</code></pre>

<p>为了简化插入操作，我们还可以创建一个触发器，当将数据插入到 &ldquo;TEST&rdquo; 表的时候，自动将最新的 ID 插入进去。</p>

<h3 id="3-创建触发器">3. 创建触发器</h3>

<pre><code>CREATE OR REPLACE TRIGGER TEST_ID_SEQ_TRG
BEFORE INSERT ON &quot;TEST&quot;
FOR EACH ROW
WHEN (NEW.&quot;ID&quot; IS NULL)
BEGIN
  SELECT TEST_ID_SEQ.NEXTVAL
  INTO :NEW.&quot;ID&quot;
  FROM DUAL;
END;
</code></pre>

<p>这样的话，每次写插入语句，只需要将 <code>ID</code> 字段的值设置为 <code>NULL</code> 它就会自动递增了：</p>

<pre><code>SQL&gt; INSERT INTO &quot;TEST&quot; (&quot;ID&quot;, &quot;NAME&quot;) VALUES (NULL, 'name4');
SQL&gt; INSERT INTO &quot;TEST&quot; (&quot;ID&quot;, &quot;NAME&quot;) VALUES (NULL, 'name5');
SQL&gt; INSERT INTO &quot;TEST&quot; (&quot;ID&quot;, &quot;NAME&quot;) VALUES (NULL, 'name6');
SQL&gt; SELECT * FROM &quot;TEST&quot;;

ID   NAME
---  ------
100	name1
101	name2
102	name3
103	name4
104	name5
105	name6
</code></pre>

<h3 id="4-一些值得注意的地方">4. 一些值得注意的地方</h3>

<h4 id="4-1-插入指定-id">4.1 插入指定 ID</h4>

<p>如果某条插入语句指定了 <code>ID</code> 的值如：</p>

<pre><code>SQL&gt; INSERT INTO &quot;TEST&quot; (&quot;ID&quot;, &quot;NAME&quot;) VALUES (1000, 'name1001');
SQL&gt; SELECT * FROM &quot;TEST&quot;;

ID   NAME
---  ------
100	name1
101	name2
102	name3
103	name4
104	name5
1000	name1001
</code></pre>

<p>那么下次 <code>ID</code> 还是会在原来的基础上继续增加：</p>

<pre><code>SQL&gt; INSERT INTO &quot;TEST&quot; (&quot;ID&quot;, &quot;NAME&quot;) VALUES (NULL, 'name1001');
SQL&gt; SELECT * FROM &quot;TEST&quot;;

ID   NAME
---  ------
100	name1
101	name2
102	name3
103	name4
104	name5
1000	name1001
</code></pre>

<p>但当序列的值到了 <code>1000</code> 的时候，如果 <code>ID</code> 允许重复，就会有两行记录 <code>ID</code> 都为 <code>1000</code>。</p>

<p>但如果 <code>ID</code> 设置为了主键，如本文的例子 <code>ID NUMBER(11) PRIMARY KEY</code>，则插入就会报错：</p>

<pre><code>Error : ORA-00001: unique constraint (SOFTWARE.SYS_C0014995) violated
</code></pre>

<h4 id="4-2-字段加引号">4.2 字段加引号</h4>

<p>在 SQL 语句中，字段最好都加上引号，不然可能会报错：</p>

<pre><code>Error : ORA-00900: invalid SQL statement
</code></pre>

<p>或：</p>

<pre><code>ORA-24344: Success with Compilation Error
</code></pre>

<h4 id="4-3-squence">4.3 SQUENCE</h4>

<ul>
<li>第一次 <code>NEXTVAL</code> 返回的是初始值；随后的 <code>NEXTVAL</code> 会自动增加 <code>INCREMENT BY</code> 对应的值，然后返回增加后的值。</li>
<li><code>CURRVAL</code> 总是返回当前 <code>SEQUENCE</code> 的值，但是在第一次 <code>NEXTVAL</code> 初始化之后才能使用 <code>CURRVAL</code> ，否则会出错。</li>
<li>一次 <code>NEXTVAL</code> 会增加一次 <code>SEQUENCE</code> 的值，所以如果在同一个语句里面使用多个NEXTVAL，其值就是不一样的。</li>
<li>如果指定 <code>CACHE</code> 值，Oracle 就可以预先在内存里面放置一些 <code>SEQUENCE</code>，这样存取的快些。 <code>CACHE</code> 里面的取完后，Oracle 自动再取一组到 <code>CACHE</code>。</li>
<li>但使用 <code>CACHE</code> 或许会跳号，比如数据库突然不正常关闭（<code>shutdown abort</code>)， <code>CACHE</code> 中的 <code>SEQUENCE</code> 就会丢失。所以可以在 <code>CREATE SEQUENCE</code> 的时候用 <code>NOCACHE</code> 防止这种情况。</li>
</ul>

<h4 id="4-4-性能">4.4 性能</h4>

<p>在数据库操作中，触发器的使用耗费系统资源相对较大。如果对于表容量相对较小的表格我们可以忽略触发器带来的性能影响。</p>

<p>考虑到大表操作的性能问题，需要尽可能的减少触发器的使用。对于以上操作，就可以抛弃触发器的使用，直接手动调用序列函数即可，但这样可能在程序维护上稍微带来一些不便。</p>

<h2 id="三-在-oracle-12c-中设置自增字段">三、在 Oracle 12c 中设置自增字段</h2>

<p>在 Oracle 12c 中设置自增字段就简单多了，因为 ORacle 12c 提供了 <code>IDENTITY</code> 属性：</p>

<pre><code>CREATE TABLE &quot;TEST&quot; (
    ID NUMBER(11) GENERATED BY DEFAULT ON NULL AS IDENTITY,
    NAME VARCHAR2(50BYTE) NOT NULL
);
</code></pre>

<p>这样就搞定了！和 MySQL 一样简单！🤣🤣🤣</p>

<h2 id="四-总结">四、总结</h2>

<p>所以如上所属，在 Oracle 中设置自增字段，需要根据不同的版本使用不同的方法：</p>

<ul>
<li>在 Oracle 11g 中，需要先创建序列（SQUENCE）再创建一个触发器（TRIGGER）。</li>
<li>在 Oracle 12c 中，只需要使用 <code>IDENTITY</code> 属性就可以了。</li>
</ul>

<hr />

<p>Github Issues <a href="https://github.com/nodejh/nodejh.github.io/issues/33">https://github.com/nodejh/nodejh.github.io/issues/33</a></p>
                </section>
            </article>

            
                <a class="twitter" href="https://twitter.com/intent/tweet?text=http%3a%2f%2fnodejh.com%2fposts%2fhow-to-create-auto-increment-column-in-oracle%2f - %e5%9c%a8%20Oracle%20%e4%b8%ad%e8%ae%be%e7%bd%ae%e8%87%aa%e5%a2%9e%e5%88%97 by @nodejh"><span class="icon-twitter"> tweet</span></a>

<a class="facebook" href="#" onclick="
    window.open(
      'https://www.facebook.com/sharer/sharer.php?u='+encodeURIComponent(location.href),
      'facebook-share-dialog',
      'width=626,height=436');
    return false;"><span class="icon-facebook-rect"> Share</span>
</a>

            

            
                <div id="disqus_thread"></div>
<script type="text/javascript">
    var disqus_shortname = 'nodejh'; 

     
    (function() {
        var dsq = document.createElement('script'); dsq.type = 'text/javascript'; dsq.async = true;
        dsq.src = '//' + disqus_shortname + '.disqus.com/embed.js';
        (document.getElementsByTagName('head')[0] || document.getElementsByTagName('body')[0]).appendChild(dsq);
    })();
</script>
<noscript>Please enable JavaScript to view the <a href="https://disqus.com/?ref_noscript">comments powered by Disqus.</a></noscript>
</div>

            

            

            <footer id="footer">
    
        <div id="social">

	
	
    <a class="symbol" href="https://www.facebook.com/nodejh">
        <i class="fa fa-facebook-square"></i>
    </a>
    
    <a class="symbol" href="https://www.github.com/nodejh">
        <i class="fa fa-github-square"></i>
    </a>
    
    <a class="symbol" href="https://www.twitter.com/nodejh">
        <i class="fa fa-twitter-square"></i>
    </a>
    


</div>

    
    <p class="small">
    
       © Copyright 2018 <i class="fa fa-heart" aria-hidden="true"></i> 
    
    </p>
    <p class="small">
        Powered by <a href="http://www.gohugo.io/">Hugo</a> Theme By <a href="https://github.com/nodejh/hugo-theme-cactus-plus">nodejh</a>
    </p>
</footer>

        </section>

        <script src="http://nodejh.com/js/jquery-2.2.4.min.js"></script>
<script src="http://nodejh.com/js/main.js"></script>
<script src="http://nodejh.com/js/highlight.min.js"></script>
<script>hljs.initHighlightingOnLoad();</script>




  
<script>
(function(i,s,o,g,r,a,m){i['GoogleAnalyticsObject']=r;i[r]=i[r]||function(){
(i[r].q=i[r].q||[]).push(arguments)},i[r].l=1*new Date();a=s.createElement(o),
m=s.getElementsByTagName(o)[0];a.async=1;a.src=g;m.parentNode.insertBefore(a,m)
})(window,document,'script','https://www.google-analytics.com/analytics.js','ga');

ga('create', 'UA-84989670-1', 'auto');
ga('send', 'pageview');
</script>





    </body>
</html>
